// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

#nullable disable

using System.IO.Compression;
using Microsoft.AspNetCore.StaticWebAssets.Tasks.Utils;
using Microsoft.Build.Framework;

namespace Microsoft.AspNetCore.StaticWebAssets.Tasks;

public class GZipCompress : Task
{
    [Required]
    public ITaskItem[] FilesToCompress { get; set; }

    public override bool Execute()
    {
        var outputDirectories = FilesToCompress
            .Select(f => Path.GetDirectoryName(f.ItemSpec))
            .Where(td => !string.IsNullOrWhiteSpace(td))
            .Distinct();

        foreach (var outputDirectory in outputDirectories)
        {
            Directory.CreateDirectory(outputDirectory);
            Log.LogMessage(MessageImportance.Low, "Created directory '{0}'.", outputDirectory);
        }

        Parallel.For(0, FilesToCompress.Length, i =>
        {
            var file = FilesToCompress[i];
            var outputRelativePath = file.ItemSpec;

            if (!AssetToCompress.TryFindInputFilePath(file, Log, out var inputFullPath))
            {
                return;
            }

            if (!File.Exists(outputRelativePath))
            {
                Log.LogMessage(MessageImportance.Low, "Compressing '{0}' because compressed file '{1}' does not exist.", inputFullPath, outputRelativePath);
            }
            else if (File.GetLastWriteTimeUtc(inputFullPath) < File.GetLastWriteTimeUtc(outputRelativePath))
            {
                // Incrementalism. If input source doesn't exist or it exists and is not newer than the expected output, do nothing.
                Log.LogMessage(MessageImportance.Low, "Skipping '{0}' because '{1}' is newer than '{2}'.", inputFullPath, outputRelativePath, inputFullPath);
                return;
            }
            else
            {
                Log.LogMessage(MessageImportance.Low, "Compressing '{0}' because file is newer than '{1}'.", inputFullPath, outputRelativePath);
            }

            try
            {
                using var sourceStream = File.OpenRead(inputFullPath);
                using var fileStream = File.Create(outputRelativePath);
                using var stream = new GZipStream(fileStream, CompressionLevel.Optimal);

                sourceStream.CopyTo(stream);
            }
            catch (Exception e)
            {
                Log.LogErrorFromException(e);
                return;
            }
        });

        return !Log.HasLoggedErrors;
    }
}
